一眼——要么0要么1
最小割!
码码码
90分(dinic
滚去写正解
还好就是建图有点烦
挂了一次是写成大根堆(这tm能过样例。
luogu2046 [NOI2010]海拔
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
using namespace std;
typedef long long LL;
typedef pair<int,int> PII;
inline LL read()
{
LL x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}
return x*f;
}
const int N=508;
int n,x;
int s,t;
int nume,head[N*N];
struct edge{
int to,nxt,f;
}e[N*N*4];
inline void add_edge(int x,int y,int z)
{
e[++nume]=(edge){y,head[x],z};head[x]=nume;
}
priority_queue<PII,vector<PII>,greater<PII> > pq;
int dis[N*N];
bool vis[N*N];
inline void dijk()
{
memset(dis,0x3f,sizeof(dis));
dis[s]=0;
pq.push(mp(0,s));
while(!pq.empty()){
int x=pq.top().second;
pq.pop();
if(vis[x]) continue;
vis[x]=1;
for(int i=head[x];i;i=e[i].nxt){
if(dis[e[i].to]>dis[x]+e[i].f){
dis[e[i].to]=dis[x]+e[i].f;
pq.push(mp(dis[e[i].to],e[i].to));
}
}
}
printf("%d",dis[t]);
}
int main()
{
n=read();
s=0;t=n*n+1;
for(int i=1;i<=n+1;++i){
for(int j=1;j<=n;++j){
x=read();
if(i==1){
add_edge(s,j,x);
}
else if(i==n+1){
add_edge(n*(n-1)+j,t,x);
}
else{
add_edge(n*(i-2)+j,n*(i-1)+j,x);
}
}
}
for(int i=1;i<=n;++i){
for(int j=1;j<=n+1;++j){
x=read();
if(j==1){
add_edge(n*(i-1)+1,t,x);
}
else if(j==n+1){
add_edge(s,n*i,x);
}
else{
add_edge(n*(i-1)+j,n*(i-1)+j-1,x);
}
}
}
for(int i=1;i<=n+1;++i){
for(int j=1;j<=n;++j){
x=read();
if(i==1){
add_edge(j,s,x);
}
else if(i==n+1){
add_edge(t,n*(n-1)+j,x);
}
else{
add_edge(n*(i-1)+j,n*(i-2)+j,x);
}
}
}
for(int i=1;i<=n;++i){
for(int j=1;j<=n+1;++j){
x=read();
if(j==1){
add_edge(t,n*(i-1)+1,x);
}
else if(j==n+1){
add_edge(n*i,s,x);
}
else{
add_edge(n*(i-1)+j-1,n*(i-1)+j,x);
}
}
}
dijk();
return 0;
}